|
1
|
|
|
/*istanbul ignore start*/'use strict'; |
|
2
|
|
|
|
|
3
|
|
|
exports.__esModule = true; |
|
4
|
|
|
exports. /*istanbul ignore end*/calcLineCount = calcLineCount; |
|
5
|
|
|
/*istanbul ignore start*/exports. /*istanbul ignore end*/merge = merge; |
|
6
|
|
|
|
|
7
|
|
|
var /*istanbul ignore start*/_create = require('./create') /*istanbul ignore end*/; |
|
8
|
|
|
|
|
9
|
|
|
var /*istanbul ignore start*/_parse = require('./parse') /*istanbul ignore end*/; |
|
10
|
|
|
|
|
11
|
|
|
var /*istanbul ignore start*/_array = require('../util/array') /*istanbul ignore end*/; |
|
12
|
|
|
|
|
13
|
|
|
/*istanbul ignore start*/function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } |
|
|
|
|
|
|
14
|
|
|
|
|
15
|
|
View Code Duplication |
/*istanbul ignore end*/function calcLineCount(hunk) { |
|
|
|
|
|
|
16
|
|
|
var conflicted = false; |
|
17
|
|
|
|
|
18
|
|
|
hunk.oldLines = 0; |
|
19
|
|
|
hunk.newLines = 0; |
|
20
|
|
|
|
|
21
|
|
|
hunk.lines.forEach(function (line) { |
|
22
|
|
|
if (typeof line !== 'string') { |
|
23
|
|
|
conflicted = true; |
|
24
|
|
|
return; |
|
25
|
|
|
} |
|
26
|
|
|
|
|
27
|
|
|
if (line[0] === '+' || line[0] === ' ') { |
|
28
|
|
|
hunk.newLines++; |
|
29
|
|
|
} |
|
30
|
|
|
if (line[0] === '-' || line[0] === ' ') { |
|
31
|
|
|
hunk.oldLines++; |
|
32
|
|
|
} |
|
33
|
|
|
}); |
|
34
|
|
|
|
|
35
|
|
|
if (conflicted) { |
|
36
|
|
|
delete hunk.oldLines; |
|
37
|
|
|
delete hunk.newLines; |
|
38
|
|
|
} |
|
39
|
|
|
} |
|
40
|
|
|
|
|
41
|
|
View Code Duplication |
function merge(mine, theirs, base) { |
|
|
|
|
|
|
42
|
|
|
mine = loadPatch(mine, base); |
|
43
|
|
|
theirs = loadPatch(theirs, base); |
|
44
|
|
|
|
|
45
|
|
|
var ret = {}; |
|
46
|
|
|
|
|
47
|
|
|
// For index we just let it pass through as it doesn't have any necessary meaning. |
|
48
|
|
|
// Leaving sanity checks on this to the API consumer that may know more about the |
|
49
|
|
|
// meaning in their own context. |
|
50
|
|
|
if (mine.index || theirs.index) { |
|
51
|
|
|
ret.index = mine.index || theirs.index; |
|
52
|
|
|
} |
|
53
|
|
|
|
|
54
|
|
|
if (mine.newFileName || theirs.newFileName) { |
|
55
|
|
|
if (!fileNameChanged(mine)) { |
|
56
|
|
|
// No header or no change in ours, use theirs (and ours if theirs does not exist) |
|
57
|
|
|
ret.oldFileName = theirs.oldFileName || mine.oldFileName; |
|
58
|
|
|
ret.newFileName = theirs.newFileName || mine.newFileName; |
|
59
|
|
|
ret.oldHeader = theirs.oldHeader || mine.oldHeader; |
|
60
|
|
|
ret.newHeader = theirs.newHeader || mine.newHeader; |
|
61
|
|
|
} else if (!fileNameChanged(theirs)) { |
|
62
|
|
|
// No header or no change in theirs, use ours |
|
63
|
|
|
ret.oldFileName = mine.oldFileName; |
|
64
|
|
|
ret.newFileName = mine.newFileName; |
|
65
|
|
|
ret.oldHeader = mine.oldHeader; |
|
66
|
|
|
ret.newHeader = mine.newHeader; |
|
67
|
|
|
} else { |
|
68
|
|
|
// Both changed... figure it out |
|
69
|
|
|
ret.oldFileName = selectField(ret, mine.oldFileName, theirs.oldFileName); |
|
70
|
|
|
ret.newFileName = selectField(ret, mine.newFileName, theirs.newFileName); |
|
71
|
|
|
ret.oldHeader = selectField(ret, mine.oldHeader, theirs.oldHeader); |
|
72
|
|
|
ret.newHeader = selectField(ret, mine.newHeader, theirs.newHeader); |
|
73
|
|
|
} |
|
74
|
|
|
} |
|
75
|
|
|
|
|
76
|
|
|
ret.hunks = []; |
|
77
|
|
|
|
|
78
|
|
|
var mineIndex = 0, |
|
79
|
|
|
theirsIndex = 0, |
|
80
|
|
|
mineOffset = 0, |
|
81
|
|
|
theirsOffset = 0; |
|
82
|
|
|
|
|
83
|
|
|
while (mineIndex < mine.hunks.length || theirsIndex < theirs.hunks.length) { |
|
84
|
|
|
var mineCurrent = mine.hunks[mineIndex] || { oldStart: Infinity }, |
|
85
|
|
|
theirsCurrent = theirs.hunks[theirsIndex] || { oldStart: Infinity }; |
|
86
|
|
|
|
|
87
|
|
|
if (hunkBefore(mineCurrent, theirsCurrent)) { |
|
88
|
|
|
// This patch does not overlap with any of the others, yay. |
|
89
|
|
|
ret.hunks.push(cloneHunk(mineCurrent, mineOffset)); |
|
90
|
|
|
mineIndex++; |
|
91
|
|
|
theirsOffset += mineCurrent.newLines - mineCurrent.oldLines; |
|
92
|
|
|
} else if (hunkBefore(theirsCurrent, mineCurrent)) { |
|
93
|
|
|
// This patch does not overlap with any of the others, yay. |
|
94
|
|
|
ret.hunks.push(cloneHunk(theirsCurrent, theirsOffset)); |
|
95
|
|
|
theirsIndex++; |
|
96
|
|
|
mineOffset += theirsCurrent.newLines - theirsCurrent.oldLines; |
|
97
|
|
|
} else { |
|
98
|
|
|
// Overlap, merge as best we can |
|
99
|
|
|
var mergedHunk = { |
|
100
|
|
|
oldStart: Math.min(mineCurrent.oldStart, theirsCurrent.oldStart), |
|
101
|
|
|
oldLines: 0, |
|
102
|
|
|
newStart: Math.min(mineCurrent.newStart + mineOffset, theirsCurrent.oldStart + theirsOffset), |
|
103
|
|
|
newLines: 0, |
|
104
|
|
|
lines: [] |
|
105
|
|
|
}; |
|
106
|
|
|
mergeLines(mergedHunk, mineCurrent.oldStart, mineCurrent.lines, theirsCurrent.oldStart, theirsCurrent.lines); |
|
107
|
|
|
theirsIndex++; |
|
108
|
|
|
mineIndex++; |
|
109
|
|
|
|
|
110
|
|
|
ret.hunks.push(mergedHunk); |
|
111
|
|
|
} |
|
112
|
|
|
} |
|
113
|
|
|
|
|
114
|
|
|
return ret; |
|
115
|
|
|
} |
|
116
|
|
|
|
|
117
|
|
|
function loadPatch(param, base) { |
|
118
|
|
|
if (typeof param === 'string') { |
|
119
|
|
|
if (/^@@/m.test(param) || /^Index:/m.test(param)) { |
|
120
|
|
|
return (/*istanbul ignore start*/(0, _parse.parsePatch) /*istanbul ignore end*/(param)[0] |
|
|
|
|
|
|
121
|
|
|
); |
|
122
|
|
|
} |
|
123
|
|
|
|
|
124
|
|
|
if (!base) { |
|
125
|
|
|
throw new Error('Must provide a base reference or pass in a patch'); |
|
126
|
|
|
} |
|
127
|
|
|
return (/*istanbul ignore start*/(0, _create.structuredPatch) /*istanbul ignore end*/(undefined, undefined, base, param) |
|
|
|
|
|
|
128
|
|
|
); |
|
129
|
|
|
} |
|
130
|
|
|
|
|
131
|
|
|
return param; |
|
132
|
|
|
} |
|
133
|
|
|
|
|
134
|
|
|
function fileNameChanged(patch) { |
|
135
|
|
|
return patch.newFileName && patch.newFileName !== patch.oldFileName; |
|
136
|
|
|
} |
|
137
|
|
|
|
|
138
|
|
|
function selectField(index, mine, theirs) { |
|
139
|
|
|
if (mine === theirs) { |
|
140
|
|
|
return mine; |
|
141
|
|
|
} else { |
|
|
|
|
|
|
142
|
|
|
index.conflict = true; |
|
143
|
|
|
return { mine: mine, theirs: theirs }; |
|
144
|
|
|
} |
|
145
|
|
|
} |
|
146
|
|
|
|
|
147
|
|
|
function hunkBefore(test, check) { |
|
148
|
|
|
return test.oldStart < check.oldStart && test.oldStart + test.oldLines < check.oldStart; |
|
149
|
|
|
} |
|
150
|
|
|
|
|
151
|
|
|
function cloneHunk(hunk, offset) { |
|
152
|
|
|
return { |
|
153
|
|
|
oldStart: hunk.oldStart, oldLines: hunk.oldLines, |
|
154
|
|
|
newStart: hunk.newStart + offset, newLines: hunk.newLines, |
|
155
|
|
|
lines: hunk.lines |
|
156
|
|
|
}; |
|
157
|
|
|
} |
|
158
|
|
|
|
|
159
|
|
View Code Duplication |
function mergeLines(hunk, mineOffset, mineLines, theirOffset, theirLines) { |
|
|
|
|
|
|
160
|
|
|
// This will generally result in a conflicted hunk, but there are cases where the context |
|
161
|
|
|
// is the only overlap where we can successfully merge the content here. |
|
162
|
|
|
var mine = { offset: mineOffset, lines: mineLines, index: 0 }, |
|
163
|
|
|
their = { offset: theirOffset, lines: theirLines, index: 0 }; |
|
164
|
|
|
|
|
165
|
|
|
// Handle any leading content |
|
166
|
|
|
insertLeading(hunk, mine, their); |
|
167
|
|
|
insertLeading(hunk, their, mine); |
|
168
|
|
|
|
|
169
|
|
|
// Now in the overlap content. Scan through and select the best changes from each. |
|
170
|
|
|
while (mine.index < mine.lines.length && their.index < their.lines.length) { |
|
171
|
|
|
var mineCurrent = mine.lines[mine.index], |
|
172
|
|
|
theirCurrent = their.lines[their.index]; |
|
173
|
|
|
|
|
174
|
|
|
if ((mineCurrent[0] === '-' || mineCurrent[0] === '+') && (theirCurrent[0] === '-' || theirCurrent[0] === '+')) { |
|
175
|
|
|
// Both modified ... |
|
176
|
|
|
mutualChange(hunk, mine, their); |
|
177
|
|
|
} else if (mineCurrent[0] === '+' && theirCurrent[0] === ' ') { |
|
178
|
|
|
/*istanbul ignore start*/var _hunk$lines; |
|
179
|
|
|
|
|
180
|
|
|
/*istanbul ignore end*/ // Mine inserted |
|
181
|
|
|
/*istanbul ignore start*/(_hunk$lines = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/collectChange(mine))); |
|
182
|
|
|
} else if (theirCurrent[0] === '+' && mineCurrent[0] === ' ') { |
|
183
|
|
|
/*istanbul ignore start*/var _hunk$lines2; |
|
184
|
|
|
|
|
185
|
|
|
/*istanbul ignore end*/ // Theirs inserted |
|
186
|
|
|
/*istanbul ignore start*/(_hunk$lines2 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines2 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/collectChange(their))); |
|
187
|
|
|
} else if (mineCurrent[0] === '-' && theirCurrent[0] === ' ') { |
|
188
|
|
|
// Mine removed or edited |
|
189
|
|
|
removal(hunk, mine, their); |
|
190
|
|
|
} else if (theirCurrent[0] === '-' && mineCurrent[0] === ' ') { |
|
191
|
|
|
// Their removed or edited |
|
192
|
|
|
removal(hunk, their, mine, true); |
|
193
|
|
|
} else if (mineCurrent === theirCurrent) { |
|
194
|
|
|
// Context identity |
|
195
|
|
|
hunk.lines.push(mineCurrent); |
|
196
|
|
|
mine.index++; |
|
197
|
|
|
their.index++; |
|
198
|
|
|
} else { |
|
199
|
|
|
// Context mismatch |
|
200
|
|
|
conflict(hunk, collectChange(mine), collectChange(their)); |
|
201
|
|
|
} |
|
202
|
|
|
} |
|
203
|
|
|
|
|
204
|
|
|
// Now push anything that may be remaining |
|
205
|
|
|
insertTrailing(hunk, mine); |
|
206
|
|
|
insertTrailing(hunk, their); |
|
207
|
|
|
|
|
208
|
|
|
calcLineCount(hunk); |
|
209
|
|
|
} |
|
210
|
|
|
|
|
211
|
|
View Code Duplication |
function mutualChange(hunk, mine, their) { |
|
|
|
|
|
|
212
|
|
|
var myChanges = collectChange(mine), |
|
213
|
|
|
theirChanges = collectChange(their); |
|
214
|
|
|
|
|
215
|
|
|
if (allRemoves(myChanges) && allRemoves(theirChanges)) { |
|
216
|
|
|
// Special case for remove changes that are supersets of one another |
|
217
|
|
|
if ( /*istanbul ignore start*/(0, _array.arrayStartsWith) /*istanbul ignore end*/(myChanges, theirChanges) && skipRemoveSuperset(their, myChanges, myChanges.length - theirChanges.length)) { |
|
|
|
|
|
|
218
|
|
|
/*istanbul ignore start*/var _hunk$lines3; |
|
219
|
|
|
|
|
220
|
|
|
/*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines3 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines3 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/myChanges)); |
|
221
|
|
|
return; |
|
222
|
|
|
} else if ( /*istanbul ignore start*/(0, _array.arrayStartsWith) /*istanbul ignore end*/(theirChanges, myChanges) && skipRemoveSuperset(mine, theirChanges, theirChanges.length - myChanges.length)) { |
|
|
|
|
|
|
223
|
|
|
/*istanbul ignore start*/var _hunk$lines4; |
|
224
|
|
|
|
|
225
|
|
|
/*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines4 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines4 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/theirChanges)); |
|
226
|
|
|
return; |
|
227
|
|
|
} |
|
228
|
|
|
} else if ( /*istanbul ignore start*/(0, _array.arrayEqual) /*istanbul ignore end*/(myChanges, theirChanges)) { |
|
|
|
|
|
|
229
|
|
|
/*istanbul ignore start*/var _hunk$lines5; |
|
230
|
|
|
|
|
231
|
|
|
/*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines5 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines5 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/myChanges)); |
|
232
|
|
|
return; |
|
233
|
|
|
} |
|
234
|
|
|
|
|
235
|
|
|
conflict(hunk, myChanges, theirChanges); |
|
236
|
|
|
} |
|
237
|
|
|
|
|
238
|
|
|
function removal(hunk, mine, their, swap) { |
|
239
|
|
|
var myChanges = collectChange(mine), |
|
240
|
|
|
theirChanges = collectContext(their, myChanges); |
|
241
|
|
|
if (theirChanges.merged) { |
|
242
|
|
|
/*istanbul ignore start*/var _hunk$lines6; |
|
243
|
|
|
|
|
244
|
|
|
/*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines6 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines6 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/theirChanges.merged)); |
|
245
|
|
|
} else { |
|
246
|
|
|
conflict(hunk, swap ? theirChanges : myChanges, swap ? myChanges : theirChanges); |
|
247
|
|
|
} |
|
248
|
|
|
} |
|
249
|
|
|
|
|
250
|
|
|
function conflict(hunk, mine, their) { |
|
251
|
|
|
hunk.conflict = true; |
|
252
|
|
|
hunk.lines.push({ |
|
253
|
|
|
conflict: true, |
|
254
|
|
|
mine: mine, |
|
255
|
|
|
theirs: their |
|
256
|
|
|
}); |
|
257
|
|
|
} |
|
258
|
|
|
|
|
259
|
|
|
function insertLeading(hunk, insert, their) { |
|
260
|
|
|
while (insert.offset < their.offset && insert.index < insert.lines.length) { |
|
261
|
|
|
var line = insert.lines[insert.index++]; |
|
262
|
|
|
hunk.lines.push(line); |
|
263
|
|
|
insert.offset++; |
|
264
|
|
|
} |
|
265
|
|
|
} |
|
266
|
|
|
function insertTrailing(hunk, insert) { |
|
267
|
|
|
while (insert.index < insert.lines.length) { |
|
268
|
|
|
var line = insert.lines[insert.index++]; |
|
269
|
|
|
hunk.lines.push(line); |
|
270
|
|
|
} |
|
271
|
|
|
} |
|
272
|
|
|
|
|
273
|
|
View Code Duplication |
function collectChange(state) { |
|
|
|
|
|
|
274
|
|
|
var ret = [], |
|
275
|
|
|
operation = state.lines[state.index][0]; |
|
276
|
|
|
while (state.index < state.lines.length) { |
|
277
|
|
|
var line = state.lines[state.index]; |
|
278
|
|
|
|
|
279
|
|
|
// Group additions that are immediately after subtractions and treat them as one "atomic" modify change. |
|
280
|
|
|
if (operation === '-' && line[0] === '+') { |
|
281
|
|
|
operation = '+'; |
|
282
|
|
|
} |
|
283
|
|
|
|
|
284
|
|
|
if (operation === line[0]) { |
|
285
|
|
|
ret.push(line); |
|
286
|
|
|
state.index++; |
|
287
|
|
|
} else { |
|
288
|
|
|
break; |
|
289
|
|
|
} |
|
290
|
|
|
} |
|
291
|
|
|
|
|
292
|
|
|
return ret; |
|
293
|
|
|
} |
|
294
|
|
View Code Duplication |
function collectContext(state, matchChanges) { |
|
|
|
|
|
|
295
|
|
|
var changes = [], |
|
296
|
|
|
merged = [], |
|
297
|
|
|
matchIndex = 0, |
|
298
|
|
|
contextChanges = false, |
|
299
|
|
|
conflicted = false; |
|
300
|
|
|
while (matchIndex < matchChanges.length && state.index < state.lines.length) { |
|
301
|
|
|
var change = state.lines[state.index], |
|
302
|
|
|
match = matchChanges[matchIndex]; |
|
303
|
|
|
|
|
304
|
|
|
// Once we've hit our add, then we are done |
|
305
|
|
|
if (match[0] === '+') { |
|
306
|
|
|
break; |
|
307
|
|
|
} |
|
308
|
|
|
|
|
309
|
|
|
contextChanges = contextChanges || change[0] !== ' '; |
|
310
|
|
|
|
|
311
|
|
|
merged.push(match); |
|
312
|
|
|
matchIndex++; |
|
313
|
|
|
|
|
314
|
|
|
// Consume any additions in the other block as a conflict to attempt |
|
315
|
|
|
// to pull in the remaining context after this |
|
316
|
|
|
if (change[0] === '+') { |
|
317
|
|
|
conflicted = true; |
|
318
|
|
|
|
|
319
|
|
|
while (change[0] === '+') { |
|
320
|
|
|
changes.push(change); |
|
321
|
|
|
change = state.lines[++state.index]; |
|
322
|
|
|
} |
|
323
|
|
|
} |
|
324
|
|
|
|
|
325
|
|
|
if (match.substr(1) === change.substr(1)) { |
|
326
|
|
|
changes.push(change); |
|
327
|
|
|
state.index++; |
|
328
|
|
|
} else { |
|
329
|
|
|
conflicted = true; |
|
330
|
|
|
} |
|
331
|
|
|
} |
|
332
|
|
|
|
|
333
|
|
|
if ((matchChanges[matchIndex] || '')[0] === '+' && contextChanges) { |
|
334
|
|
|
conflicted = true; |
|
335
|
|
|
} |
|
336
|
|
|
|
|
337
|
|
|
if (conflicted) { |
|
338
|
|
|
return changes; |
|
339
|
|
|
} |
|
340
|
|
|
|
|
341
|
|
|
while (matchIndex < matchChanges.length) { |
|
342
|
|
|
merged.push(matchChanges[matchIndex++]); |
|
343
|
|
|
} |
|
344
|
|
|
|
|
345
|
|
|
return { |
|
346
|
|
|
merged: merged, |
|
347
|
|
|
changes: changes |
|
348
|
|
|
}; |
|
349
|
|
|
} |
|
350
|
|
|
|
|
351
|
|
|
function allRemoves(changes) { |
|
352
|
|
|
return changes.reduce(function (prev, change) { |
|
353
|
|
|
return prev && change[0] === '-'; |
|
354
|
|
|
}, true); |
|
355
|
|
|
} |
|
356
|
|
|
function skipRemoveSuperset(state, removeChanges, delta) { |
|
357
|
|
|
for (var i = 0; i < delta; i++) { |
|
358
|
|
|
var changeContent = removeChanges[removeChanges.length - delta + i].substr(1); |
|
359
|
|
|
if (state.lines[state.index + i] !== ' ' + changeContent) { |
|
360
|
|
|
return false; |
|
361
|
|
|
} |
|
362
|
|
|
} |
|
363
|
|
|
|
|
364
|
|
|
state.index += delta; |
|
365
|
|
|
return true; |
|
366
|
|
|
} |
|
367
|
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/patch/merge.js"],"names":["calcLineCount","merge","hunk","conflicted","oldLines","newLines","lines","forEach","line","mine","theirs","base","loadPatch","ret","index","newFileName","fileNameChanged","oldFileName","oldHeader","newHeader","selectField","hunks","mineIndex","theirsIndex","mineOffset","theirsOffset","length","mineCurrent","oldStart","Infinity","theirsCurrent","hunkBefore","push","cloneHunk","mergedHunk","Math","min","newStart","mergeLines","param","test","Error","undefined","patch","conflict","check","offset","mineLines","theirOffset","theirLines","their","insertLeading","theirCurrent","mutualChange","collectChange","removal","insertTrailing","myChanges","theirChanges","allRemoves","skipRemoveSuperset","swap","collectContext","merged","insert","state","operation","matchChanges","changes","matchIndex","contextChanges","change","match","substr","reduce","prev","removeChanges","delta","i","changeContent"],"mappings":";;;gCAKgBA,a,GAAAA,a;yDA0BAC,K,GAAAA,K;;AA/BhB;;AACA;;AAEA;;;;uBAEO,SAASD,aAAT,CAAuBE,IAAvB,EAA6B;AAClC,MAAIC,aAAa,KAAjB;;AAEAD,OAAKE,QAAL,GAAgB,CAAhB;AACAF,OAAKG,QAAL,GAAgB,CAAhB;;AAEAH,OAAKI,KAAL,CAAWC,OAAX,CAAmB,UAASC,IAAT,EAAe;AAChC,QAAI,OAAOA,IAAP,KAAgB,QAApB,EAA8B;AAC5BL,mBAAa,IAAb;AACA;AACD;;AAED,QAAIK,KAAK,CAAL,MAAY,GAAZ,IAAmBA,KAAK,CAAL,MAAY,GAAnC,EAAwC;AACtCN,WAAKG,QAAL;AACD;AACD,QAAIG,KAAK,CAAL,MAAY,GAAZ,IAAmBA,KAAK,CAAL,MAAY,GAAnC,EAAwC;AACtCN,WAAKE,QAAL;AACD;AACF,GAZD;;AAcA,MAAID,UAAJ,EAAgB;AACd,WAAOD,KAAKE,QAAZ;AACA,WAAOF,KAAKG,QAAZ;AACD;AACF;;AAEM,SAASJ,KAAT,CAAeQ,IAAf,EAAqBC,MAArB,EAA6BC,IAA7B,EAAmC;AACxCF,SAAOG,UAAUH,IAAV,EAAgBE,IAAhB,CAAP;AACAD,WAASE,UAAUF,MAAV,EAAkBC,IAAlB,CAAT;;AAEA,MAAIE,MAAM,EAAV;;AAEA;AACA;AACA;AACA,MAAIJ,KAAKK,KAAL,IAAcJ,OAAOI,KAAzB,EAAgC;AAC9BD,QAAIC,KAAJ,GAAYL,KAAKK,KAAL,IAAcJ,OAAOI,KAAjC;AACD;;AAED,MAAIL,KAAKM,WAAL,IAAoBL,OAAOK,WAA/B,EAA4C;AAC1C,QAAI,CAACC,gBAAgBP,IAAhB,CAAL,EAA4B;AAC1B;AACAI,UAAII,WAAJ,GAAkBP,OAAOO,WAAP,IAAsBR,KAAKQ,WAA7C;AACAJ,UAAIE,WAAJ,GAAkBL,OAAOK,WAAP,IAAsBN,KAAKM,WAA7C;AACAF,UAAIK,SAAJ,GAAgBR,OAAOQ,SAAP,IAAoBT,KAAKS,SAAzC;AACAL,UAAIM,SAAJ,GAAgBT,OAAOS,SAAP,IAAoBV,KAAKU,SAAzC;AACD,KAND,MAMO,IAAI,CAACH,gBAAgBN,MAAhB,CAAL,EAA8B;AACnC;AACAG,UAAII,WAAJ,GAAkBR,KAAKQ,WAAvB;AACAJ,UAAIE,WAAJ,GAAkBN,KAAKM,WAAvB;AACAF,UAAIK,SAAJ,GAAgBT,KAAKS,SAArB;AACAL,UAAIM,SAAJ,GAAgBV,KAAKU,SAArB;AACD,KANM,MAMA;AACL;AACAN,UAAII,WAAJ,GAAkBG,YAAYP,GAAZ,EAAiBJ,KAAKQ,WAAtB,EAAmCP,OAAOO,WAA1C,CAAlB;AACAJ,UAAIE,WAAJ,GAAkBK,YAAYP,GAAZ,EAAiBJ,KAAKM,WAAtB,EAAmCL,OAAOK,WAA1C,CAAlB;AACAF,UAAIK,SAAJ,GAAgBE,YAAYP,GAAZ,EAAiBJ,KAAKS,SAAtB,EAAiCR,OAAOQ,SAAxC,CAAhB;AACAL,UAAIM,SAAJ,GAAgBC,YAAYP,GAAZ,EAAiBJ,KAAKU,SAAtB,EAAiCT,OAAOS,SAAxC,CAAhB;AACD;AACF;;AAEDN,MAAIQ,KAAJ,GAAY,EAAZ;;AAEA,MAAIC,YAAY,CAAhB;AAAA,MACIC,cAAc,CADlB;AAAA,MAEIC,aAAa,CAFjB;AAAA,MAGIC,eAAe,CAHnB;;AAKA,SAAOH,YAAYb,KAAKY,KAAL,CAAWK,MAAvB,IAAiCH,cAAcb,OAAOW,KAAP,CAAaK,MAAnE,EAA2E;AACzE,QAAIC,cAAclB,KAAKY,KAAL,CAAWC,SAAX,KAAyB,EAACM,UAAUC,QAAX,EAA3C;AAAA,QACIC,gBAAgBpB,OAAOW,KAAP,CAAaE,WAAb,KAA6B,EAACK,UAAUC,QAAX,EADjD;;AAGA,QAAIE,WAAWJ,WAAX,EAAwBG,aAAxB,CAAJ,EAA4C;AAC1C;AACAjB,UAAIQ,KAAJ,CAAUW,IAAV,CAAeC,UAAUN,WAAV,EAAuBH,UAAvB,CAAf;AACAF;AACAG,sBAAgBE,YAAYtB,QAAZ,GAAuBsB,YAAYvB,QAAnD;AACD,KALD,MAKO,IAAI2B,WAAWD,aAAX,EAA0BH,WAA1B,CAAJ,EAA4C;AACjD;AACAd,UAAIQ,KAAJ,CAAUW,IAAV,CAAeC,UAAUH,aAAV,EAAyBL,YAAzB,CAAf;AACAF;AACAC,oBAAcM,cAAczB,QAAd,GAAyByB,cAAc1B,QAArD;AACD,KALM,MAKA;AACL;AACA,UAAI8B,aAAa;AACfN,kBAAUO,KAAKC,GAAL,CAAST,YAAYC,QAArB,EAA+BE,cAAcF,QAA7C,CADK;AAEfxB,kBAAU,CAFK;AAGfiC,kBAAUF,KAAKC,GAAL,CAAST,YAAYU,QAAZ,GAAuBb,UAAhC,EAA4CM,cAAcF,QAAd,GAAyBH,YAArE,CAHK;AAIfpB,kBAAU,CAJK;AAKfC,eAAO;AALQ,OAAjB;AAOAgC,iBAAWJ,UAAX,EAAuBP,YAAYC,QAAnC,EAA6CD,YAAYrB,KAAzD,EAAgEwB,cAAcF,QAA9E,EAAwFE,cAAcxB,KAAtG;AACAiB;AACAD;;AAEAT,UAAIQ,KAAJ,CAAUW,IAAV,CAAeE,UAAf;AACD;AACF;;AAED,SAAOrB,GAAP;AACD;;AAED,SAASD,SAAT,CAAmB2B,KAAnB,EAA0B5B,IAA1B,EAAgC;AAC9B,MAAI,OAAO4B,KAAP,KAAiB,QAArB,EAA+B;AAC7B,QAAI,OAAOC,IAAP,CAAYD,KAAZ,KAAuB,WAAWC,IAAX,CAAgBD,KAAhB,CAA3B,EAAoD;AAClD,aAAO,yEAAWA,KAAX,EAAkB,CAAlB;AAAP;AACD;;AAED,QAAI,CAAC5B,IAAL,EAAW;AACT,YAAM,IAAI8B,KAAJ,CAAU,kDAAV,CAAN;AACD;AACD,WAAO,+EAAgBC,SAAhB,EAA2BA,SAA3B,EAAsC/B,IAAtC,EAA4C4B,KAA5C;AAAP;AACD;;AAED,SAAOA,KAAP;AACD;;AAED,SAASvB,eAAT,CAAyB2B,KAAzB,EAAgC;AAC9B,SAAOA,MAAM5B,WAAN,IAAqB4B,MAAM5B,WAAN,KAAsB4B,MAAM1B,WAAxD;AACD;;AAED,SAASG,WAAT,CAAqBN,KAArB,EAA4BL,IAA5B,EAAkCC,MAAlC,EAA0C;AACxC,MAAID,SAASC,MAAb,EAAqB;AACnB,WAAOD,IAAP;AACD,GAFD,MAEO;AACLK,UAAM8B,QAAN,GAAiB,IAAjB;AACA,WAAO,EAACnC,UAAD,EAAOC,cAAP,EAAP;AACD;AACF;;AAED,SAASqB,UAAT,CAAoBS,IAApB,EAA0BK,KAA1B,EAAiC;AAC/B,SAAOL,KAAKZ,QAAL,GAAgBiB,MAAMjB,QAAtB,IACDY,KAAKZ,QAAL,GAAgBY,KAAKpC,QAAtB,GAAkCyC,MAAMjB,QAD7C;AAED;;AAED,SAASK,SAAT,CAAmB/B,IAAnB,EAAyB4C,MAAzB,EAAiC;AAC/B,SAAO;AACLlB,cAAU1B,KAAK0B,QADV,EACoBxB,UAAUF,KAAKE,QADnC;AAELiC,cAAUnC,KAAKmC,QAAL,GAAgBS,MAFrB,EAE6BzC,UAAUH,KAAKG,QAF5C;AAGLC,WAAOJ,KAAKI;AAHP,GAAP;AAKD;;AAED,SAASgC,UAAT,CAAoBpC,IAApB,EAA0BsB,UAA1B,EAAsCuB,SAAtC,EAAiDC,WAAjD,EAA8DC,UAA9D,EAA0E;AACxE;AACA;AACA,MAAIxC,OAAO,EAACqC,QAAQtB,UAAT,EAAqBlB,OAAOyC,SAA5B,EAAuCjC,OAAO,CAA9C,EAAX;AAAA,MACIoC,QAAQ,EAACJ,QAAQE,WAAT,EAAsB1C,OAAO2C,UAA7B,EAAyCnC,OAAO,CAAhD,EADZ;;AAGA;AACAqC,gBAAcjD,IAAd,EAAoBO,IAApB,EAA0ByC,KAA1B;AACAC,gBAAcjD,IAAd,EAAoBgD,KAApB,EAA2BzC,IAA3B;;AAEA;AACA,SAAOA,KAAKK,KAAL,GAAaL,KAAKH,KAAL,CAAWoB,MAAxB,IAAkCwB,MAAMpC,KAAN,GAAcoC,MAAM5C,KAAN,CAAYoB,MAAnE,EAA2E;AACzE,QAAIC,cAAclB,KAAKH,KAAL,CAAWG,KAAKK,KAAhB,CAAlB;AAAA,QACIsC,eAAeF,MAAM5C,KAAN,CAAY4C,MAAMpC,KAAlB,CADnB;;AAGA,QAAI,CAACa,YAAY,CAAZ,MAAmB,GAAnB,IAA0BA,YAAY,CAAZ,MAAmB,GAA9C,MACIyB,aAAa,CAAb,MAAoB,GAApB,IAA2BA,aAAa,CAAb,MAAoB,GADnD,CAAJ,EAC6D;AAC3D;AACAC,mBAAanD,IAAb,EAAmBO,IAAnB,EAAyByC,KAAzB;AACD,KAJD,MAIO,IAAIvB,YAAY,CAAZ,MAAmB,GAAnB,IAA0ByB,aAAa,CAAb,MAAoB,GAAlD,EAAuD;AAAA;;AAAA,8BAC5D;AACA,0EAAK9C,KAAL,EAAW0B,IAAX,4LAAoBsB,cAAc7C,IAAd,CAApB;AACD,KAHM,MAGA,IAAI2C,aAAa,CAAb,MAAoB,GAApB,IAA2BzB,YAAY,CAAZ,MAAmB,GAAlD,EAAuD;AAAA;;AAAA,8BAC5D;AACA,2EAAKrB,KAAL,EAAW0B,IAAX,6LAAoBsB,cAAcJ,KAAd,CAApB;AACD,KAHM,MAGA,IAAIvB,YAAY,CAAZ,MAAmB,GAAnB,IAA0ByB,aAAa,CAAb,MAAoB,GAAlD,EAAuD;AAC5D;AACAG,cAAQrD,IAAR,EAAcO,IAAd,EAAoByC,KAApB;AACD,KAHM,MAGA,IAAIE,aAAa,CAAb,MAAoB,GAApB,IAA2BzB,YAAY,CAAZ,MAAmB,GAAlD,EAAuD;AAC5D;AACA4B,cAAQrD,IAAR,EAAcgD,KAAd,EAAqBzC,IAArB,EAA2B,IAA3B;AACD,KAHM,MAGA,IAAIkB,gBAAgByB,YAApB,EAAkC;AACvC;AACAlD,WAAKI,KAAL,CAAW0B,IAAX,CAAgBL,WAAhB;AACAlB,WAAKK,KAAL;AACAoC,YAAMpC,KAAN;AACD,KALM,MAKA;AACL;AACA8B,eAAS1C,IAAT,EAAeoD,cAAc7C,IAAd,CAAf,EAAoC6C,cAAcJ,KAAd,CAApC;AACD;AACF;;AAED;AACAM,iBAAetD,IAAf,EAAqBO,IAArB;AACA+C,iBAAetD,IAAf,EAAqBgD,KAArB;;AAEAlD,gBAAcE,IAAd;AACD;;AAED,SAASmD,YAAT,CAAsBnD,IAAtB,EAA4BO,IAA5B,EAAkCyC,KAAlC,EAAyC;AACvC,MAAIO,YAAYH,cAAc7C,IAAd,CAAhB;AAAA,MACIiD,eAAeJ,cAAcJ,KAAd,CADnB;;AAGA,MAAIS,WAAWF,SAAX,KAAyBE,WAAWD,YAAX,CAA7B,EAAuD;AACrD;AACA,QAAI,8EAAgBD,SAAhB,EAA2BC,YAA3B,KACGE,mBAAmBV,KAAnB,EAA0BO,SAA1B,EAAqCA,UAAU/B,MAAV,GAAmBgC,aAAahC,MAArE,CADP,EACqF;AAAA;;AAAA,6BACnF,sEAAKpB,KAAL,EAAW0B,IAAX,6LAAoByB,SAApB;AACA;AACD,KAJD,MAIO,IAAI,8EAAgBC,YAAhB,EAA8BD,SAA9B,KACJG,mBAAmBnD,IAAnB,EAAyBiD,YAAzB,EAAuCA,aAAahC,MAAb,GAAsB+B,UAAU/B,MAAvE,CADA,EACgF;AAAA;;AAAA,6BACrF,sEAAKpB,KAAL,EAAW0B,IAAX,6LAAoB0B,YAApB;AACA;AACD;AACF,GAXD,MAWO,IAAI,yEAAWD,SAAX,EAAsBC,YAAtB,CAAJ,EAAyC;AAAA;;AAAA,2BAC9C,sEAAKpD,KAAL,EAAW0B,IAAX,6LAAoByB,SAApB;AACA;AACD;;AAEDb,WAAS1C,IAAT,EAAeuD,SAAf,EAA0BC,YAA1B;AACD;;AAED,SAASH,OAAT,CAAiBrD,IAAjB,EAAuBO,IAAvB,EAA6ByC,KAA7B,EAAoCW,IAApC,EAA0C;AACxC,MAAIJ,YAAYH,cAAc7C,IAAd,CAAhB;AAAA,MACIiD,eAAeI,eAAeZ,KAAf,EAAsBO,SAAtB,CADnB;AAEA,MAAIC,aAAaK,MAAjB,EAAyB;AAAA;;AAAA,2BACvB,sEAAKzD,KAAL,EAAW0B,IAAX,6LAAoB0B,aAAaK,MAAjC;AACD,GAFD,MAEO;AACLnB,aAAS1C,IAAT,EAAe2D,OAAOH,YAAP,GAAsBD,SAArC,EAAgDI,OAAOJ,SAAP,GAAmBC,YAAnE;AACD;AACF;;AAED,SAASd,QAAT,CAAkB1C,IAAlB,EAAwBO,IAAxB,EAA8ByC,KAA9B,EAAqC;AACnChD,OAAK0C,QAAL,GAAgB,IAAhB;AACA1C,OAAKI,KAAL,CAAW0B,IAAX,CAAgB;AACdY,cAAU,IADI;AAEdnC,UAAMA,IAFQ;AAGdC,YAAQwC;AAHM,GAAhB;AAKD;;AAED,SAASC,aAAT,CAAuBjD,IAAvB,EAA6B8D,MAA7B,EAAqCd,KAArC,EAA4C;AAC1C,SAAOc,OAAOlB,MAAP,GAAgBI,MAAMJ,MAAtB,IAAgCkB,OAAOlD,KAAP,GAAekD,OAAO1D,KAAP,CAAaoB,MAAnE,EAA2E;AACzE,QAAIlB,OAAOwD,OAAO1D,KAAP,CAAa0D,OAAOlD,KAAP,EAAb,CAAX;AACAZ,SAAKI,KAAL,CAAW0B,IAAX,CAAgBxB,IAAhB;AACAwD,WAAOlB,MAAP;AACD;AACF;AACD,SAASU,cAAT,CAAwBtD,IAAxB,EAA8B8D,MAA9B,EAAsC;AACpC,SAAOA,OAAOlD,KAAP,GAAekD,OAAO1D,KAAP,CAAaoB,MAAnC,EAA2C;AACzC,QAAIlB,OAAOwD,OAAO1D,KAAP,CAAa0D,OAAOlD,KAAP,EAAb,CAAX;AACAZ,SAAKI,KAAL,CAAW0B,IAAX,CAAgBxB,IAAhB;AACD;AACF;;AAED,SAAS8C,aAAT,CAAuBW,KAAvB,EAA8B;AAC5B,MAAIpD,MAAM,EAAV;AAAA,MACIqD,YAAYD,MAAM3D,KAAN,CAAY2D,MAAMnD,KAAlB,EAAyB,CAAzB,CADhB;AAEA,SAAOmD,MAAMnD,KAAN,GAAcmD,MAAM3D,KAAN,CAAYoB,MAAjC,EAAyC;AACvC,QAAIlB,OAAOyD,MAAM3D,KAAN,CAAY2D,MAAMnD,KAAlB,CAAX;;AAEA;AACA,QAAIoD,cAAc,GAAd,IAAqB1D,KAAK,CAAL,MAAY,GAArC,EAA0C;AACxC0D,kBAAY,GAAZ;AACD;;AAED,QAAIA,cAAc1D,KAAK,CAAL,CAAlB,EAA2B;AACzBK,UAAImB,IAAJ,CAASxB,IAAT;AACAyD,YAAMnD,KAAN;AACD,KAHD,MAGO;AACL;AACD;AACF;;AAED,SAAOD,GAAP;AACD;AACD,SAASiD,cAAT,CAAwBG,KAAxB,EAA+BE,YAA/B,EAA6C;AAC3C,MAAIC,UAAU,EAAd;AAAA,MACIL,SAAS,EADb;AAAA,MAEIM,aAAa,CAFjB;AAAA,MAGIC,iBAAiB,KAHrB;AAAA,MAIInE,aAAa,KAJjB;AAKA,SAAOkE,aAAaF,aAAazC,MAA1B,IACEuC,MAAMnD,KAAN,GAAcmD,MAAM3D,KAAN,CAAYoB,MADnC,EAC2C;AACzC,QAAI6C,SAASN,MAAM3D,KAAN,CAAY2D,MAAMnD,KAAlB,CAAb;AAAA,QACI0D,QAAQL,aAAaE,UAAb,CADZ;;AAGA;AACA,QAAIG,MAAM,CAAN,MAAa,GAAjB,EAAsB;AACpB;AACD;;AAEDF,qBAAiBA,kBAAkBC,OAAO,CAAP,MAAc,GAAjD;;AAEAR,WAAO/B,IAAP,CAAYwC,KAAZ;AACAH;;AAEA;AACA;AACA,QAAIE,OAAO,CAAP,MAAc,GAAlB,EAAuB;AACrBpE,mBAAa,IAAb;;AAEA,aAAOoE,OAAO,CAAP,MAAc,GAArB,EAA0B;AACxBH,gBAAQpC,IAAR,CAAauC,MAAb;AACAA,iBAASN,MAAM3D,KAAN,CAAY,EAAE2D,MAAMnD,KAApB,CAAT;AACD;AACF;;AAED,QAAI0D,MAAMC,MAAN,CAAa,CAAb,MAAoBF,OAAOE,MAAP,CAAc,CAAd,CAAxB,EAA0C;AACxCL,cAAQpC,IAAR,CAAauC,MAAb;AACAN,YAAMnD,KAAN;AACD,KAHD,MAGO;AACLX,mBAAa,IAAb;AACD;AACF;;AAED,MAAI,CAACgE,aAAaE,UAAb,KAA4B,EAA7B,EAAiC,CAAjC,MAAwC,GAAxC,IACGC,cADP,EACuB;AACrBnE,iBAAa,IAAb;AACD;;AAED,MAAIA,UAAJ,EAAgB;AACd,WAAOiE,OAAP;AACD;;AAED,SAAOC,aAAaF,aAAazC,MAAjC,EAAyC;AACvCqC,WAAO/B,IAAP,CAAYmC,aAAaE,YAAb,CAAZ;AACD;;AAED,SAAO;AACLN,kBADK;AAELK;AAFK,GAAP;AAID;;AAED,SAAST,UAAT,CAAoBS,OAApB,EAA6B;AAC3B,SAAOA,QAAQM,MAAR,CAAe,UAASC,IAAT,EAAeJ,MAAf,EAAuB;AAC3C,WAAOI,QAAQJ,OAAO,CAAP,MAAc,GAA7B;AACD,GAFM,EAEJ,IAFI,CAAP;AAGD;AACD,SAASX,kBAAT,CAA4BK,KAA5B,EAAmCW,aAAnC,EAAkDC,KAAlD,EAAyD;AACvD,OAAK,IAAIC,IAAI,CAAb,EAAgBA,IAAID,KAApB,EAA2BC,GAA3B,EAAgC;AAC9B,QAAIC,gBAAgBH,cAAcA,cAAclD,MAAd,GAAuBmD,KAAvB,GAA+BC,CAA7C,EAAgDL,MAAhD,CAAuD,CAAvD,CAApB;AACA,QAAIR,MAAM3D,KAAN,CAAY2D,MAAMnD,KAAN,GAAcgE,CAA1B,MAAiC,MAAMC,aAA3C,EAA0D;AACxD,aAAO,KAAP;AACD;AACF;;AAEDd,QAAMnD,KAAN,IAAe+D,KAAf;AACA,SAAO,IAAP;AACD","file":"merge.js","sourcesContent":["import {structuredPatch} from './create';\nimport {parsePatch} from './parse';\n\nimport {arrayEqual, arrayStartsWith} from '../util/array';\n\nexport function calcLineCount(hunk) {\n  let conflicted = false;\n\n  hunk.oldLines = 0;\n  hunk.newLines = 0;\n\n  hunk.lines.forEach(function(line) {\n    if (typeof line !== 'string') {\n      conflicted = true;\n      return;\n    }\n\n    if (line[0] === '+' || line[0] === ' ') {\n      hunk.newLines++;\n    }\n    if (line[0] === '-' || line[0] === ' ') {\n      hunk.oldLines++;\n    }\n  });\n\n  if (conflicted) {\n    delete hunk.oldLines;\n    delete hunk.newLines;\n  }\n}\n\nexport function merge(mine, theirs, base) {\n  mine = loadPatch(mine, base);\n  theirs = loadPatch(theirs, base);\n\n  let ret = {};\n\n  // For index we just let it pass through as it doesn't have any necessary meaning.\n  // Leaving sanity checks on this to the API consumer that may know more about the\n  // meaning in their own context.\n  if (mine.index || theirs.index) {\n    ret.index = mine.index || theirs.index;\n  }\n\n  if (mine.newFileName || theirs.newFileName) {\n    if (!fileNameChanged(mine)) {\n      // No header or no change in ours, use theirs (and ours if theirs does not exist)\n      ret.oldFileName = theirs.oldFileName || mine.oldFileName;\n      ret.newFileName = theirs.newFileName || mine.newFileName;\n      ret.oldHeader = theirs.oldHeader || mine.oldHeader;\n      ret.newHeader = theirs.newHeader || mine.newHeader;\n    } else if (!fileNameChanged(theirs)) {\n      // No header or no change in theirs, use ours\n      ret.oldFileName = mine.oldFileName;\n      ret.newFileName = mine.newFileName;\n      ret.oldHeader = mine.oldHeader;\n      ret.newHeader = mine.newHeader;\n    } else {\n      // Both changed... figure it out\n      ret.oldFileName = selectField(ret, mine.oldFileName, theirs.oldFileName);\n      ret.newFileName = selectField(ret, mine.newFileName, theirs.newFileName);\n      ret.oldHeader = selectField(ret, mine.oldHeader, theirs.oldHeader);\n      ret.newHeader = selectField(ret, mine.newHeader, theirs.newHeader);\n    }\n  }\n\n  ret.hunks = [];\n\n  let mineIndex = 0,\n      theirsIndex = 0,\n      mineOffset = 0,\n      theirsOffset = 0;\n\n  while (mineIndex < mine.hunks.length || theirsIndex < theirs.hunks.length) {\n    let mineCurrent = mine.hunks[mineIndex] || {oldStart: Infinity},\n        theirsCurrent = theirs.hunks[theirsIndex] || {oldStart: Infinity};\n\n    if (hunkBefore(mineCurrent, theirsCurrent)) {\n      // This patch does not overlap with any of the others, yay.\n      ret.hunks.push(cloneHunk(mineCurrent, mineOffset));\n      mineIndex++;\n      theirsOffset += mineCurrent.newLines - mineCurrent.oldLines;\n    } else if (hunkBefore(theirsCurrent, mineCurrent)) {\n      // This patch does not overlap with any of the others, yay.\n      ret.hunks.push(cloneHunk(theirsCurrent, theirsOffset));\n      theirsIndex++;\n      mineOffset += theirsCurrent.newLines - theirsCurrent.oldLines;\n    } else {\n      // Overlap, merge as best we can\n      let mergedHunk = {\n        oldStart: Math.min(mineCurrent.oldStart, theirsCurrent.oldStart),\n        oldLines: 0,\n        newStart: Math.min(mineCurrent.newStart + mineOffset, theirsCurrent.oldStart + theirsOffset),\n        newLines: 0,\n        lines: []\n      };\n      mergeLines(mergedHunk, mineCurrent.oldStart, mineCurrent.lines, theirsCurrent.oldStart, theirsCurrent.lines);\n      theirsIndex++;\n      mineIndex++;\n\n      ret.hunks.push(mergedHunk);\n    }\n  }\n\n  return ret;\n}\n\nfunction loadPatch(param, base) {\n  if (typeof param === 'string') {\n    if (/^@@/m.test(param) || (/^Index:/m.test(param))) {\n      return parsePatch(param)[0];\n    }\n\n    if (!base) {\n      throw new Error('Must provide a base reference or pass in a patch');\n    }\n    return structuredPatch(undefined, undefined, base, param);\n  }\n\n  return param;\n}\n\nfunction fileNameChanged(patch) {\n  return patch.newFileName && patch.newFileName !== patch.oldFileName;\n}\n\nfunction selectField(index, mine, theirs) {\n  if (mine === theirs) {\n    return mine;\n  } else {\n    index.conflict = true;\n    return {mine, theirs};\n  }\n}\n\nfunction hunkBefore(test, check) {\n  return test.oldStart < check.oldStart\n    && (test.oldStart + test.oldLines) < check.oldStart;\n}\n\nfunction cloneHunk(hunk, offset) {\n  return {\n    oldStart: hunk.oldStart, oldLines: hunk.oldLines,\n    newStart: hunk.newStart + offset, newLines: hunk.newLines,\n    lines: hunk.lines\n  };\n}\n\nfunction mergeLines(hunk, mineOffset, mineLines, theirOffset, theirLines) {\n  // This will generally result in a conflicted hunk, but there are cases where the context\n  // is the only overlap where we can successfully merge the content here.\n  let mine = {offset: mineOffset, lines: mineLines, index: 0},\n      their = {offset: theirOffset, lines: theirLines, index: 0};\n\n  // Handle any leading content\n  insertLeading(hunk, mine, their);\n  insertLeading(hunk, their, mine);\n\n  // Now in the overlap content. Scan through and select the best changes from each.\n  while (mine.index < mine.lines.length && their.index < their.lines.length) {\n    let mineCurrent = mine.lines[mine.index],\n        theirCurrent = their.lines[their.index];\n\n    if ((mineCurrent[0] === '-' || mineCurrent[0] === '+')\n        && (theirCurrent[0] === '-' || theirCurrent[0] === '+')) {\n      // Both modified ...\n      mutualChange(hunk, mine, their);\n    } else if (mineCurrent[0] === '+' && theirCurrent[0] === ' ') {\n      // Mine inserted\n      hunk.lines.push(... collectChange(mine));\n    } else if (theirCurrent[0] === '+' && mineCurrent[0] === ' ') {\n      // Theirs inserted\n      hunk.lines.push(... collectChange(their));\n    } else if (mineCurrent[0] === '-' && theirCurrent[0] === ' ') {\n      // Mine removed or edited\n      removal(hunk, mine, their);\n    } else if (theirCurrent[0] === '-' && mineCurrent[0] === ' ') {\n      // Their removed or edited\n      removal(hunk, their, mine, true);\n    } else if (mineCurrent === theirCurrent) {\n      // Context identity\n      hunk.lines.push(mineCurrent);\n      mine.index++;\n      their.index++;\n    } else {\n      // Context mismatch\n      conflict(hunk, collectChange(mine), collectChange(their));\n    }\n  }\n\n  // Now push anything that may be remaining\n  insertTrailing(hunk, mine);\n  insertTrailing(hunk, their);\n\n  calcLineCount(hunk);\n}\n\nfunction mutualChange(hunk, mine, their) {\n  let myChanges = collectChange(mine),\n      theirChanges = collectChange(their);\n\n  if (allRemoves(myChanges) && allRemoves(theirChanges)) {\n    // Special case for remove changes that are supersets of one another\n    if (arrayStartsWith(myChanges, theirChanges)\n        && skipRemoveSuperset(their, myChanges, myChanges.length - theirChanges.length)) {\n      hunk.lines.push(... myChanges);\n      return;\n    } else if (arrayStartsWith(theirChanges, myChanges)\n        && skipRemoveSuperset(mine, theirChanges, theirChanges.length - myChanges.length)) {\n      hunk.lines.push(... theirChanges);\n      return;\n    }\n  } else if (arrayEqual(myChanges, theirChanges)) {\n    hunk.lines.push(... myChanges);\n    return;\n  }\n\n  conflict(hunk, myChanges, theirChanges);\n}\n\nfunction removal(hunk, mine, their, swap) {\n  let myChanges = collectChange(mine),\n      theirChanges = collectContext(their, myChanges);\n  if (theirChanges.merged) {\n    hunk.lines.push(... theirChanges.merged);\n  } else {\n    conflict(hunk, swap ? theirChanges : myChanges, swap ? myChanges : theirChanges);\n  }\n}\n\nfunction conflict(hunk, mine, their) {\n  hunk.conflict = true;\n  hunk.lines.push({\n    conflict: true,\n    mine: mine,\n    theirs: their\n  });\n}\n\nfunction insertLeading(hunk, insert, their) {\n  while (insert.offset < their.offset && insert.index < insert.lines.length) {\n    let line = insert.lines[insert.index++];\n    hunk.lines.push(line);\n    insert.offset++;\n  }\n}\nfunction insertTrailing(hunk, insert) {\n  while (insert.index < insert.lines.length) {\n    let line = insert.lines[insert.index++];\n    hunk.lines.push(line);\n  }\n}\n\nfunction collectChange(state) {\n  let ret = [],\n      operation = state.lines[state.index][0];\n  while (state.index < state.lines.length) {\n    let line = state.lines[state.index];\n\n    // Group additions that are immediately after subtractions and treat them as one \"atomic\" modify change.\n    if (operation === '-' && line[0] === '+') {\n      operation = '+';\n    }\n\n    if (operation === line[0]) {\n      ret.push(line);\n      state.index++;\n    } else {\n      break;\n    }\n  }\n\n  return ret;\n}\nfunction collectContext(state, matchChanges) {\n  let changes = [],\n      merged = [],\n      matchIndex = 0,\n      contextChanges = false,\n      conflicted = false;\n  while (matchIndex < matchChanges.length\n        && state.index < state.lines.length) {\n    let change = state.lines[state.index],\n        match = matchChanges[matchIndex];\n\n    // Once we've hit our add, then we are done\n    if (match[0] === '+') {\n      break;\n    }\n\n    contextChanges = contextChanges || change[0] !== ' ';\n\n    merged.push(match);\n    matchIndex++;\n\n    // Consume any additions in the other block as a conflict to attempt\n    // to pull in the remaining context after this\n    if (change[0] === '+') {\n      conflicted = true;\n\n      while (change[0] === '+') {\n        changes.push(change);\n        change = state.lines[++state.index];\n      }\n    }\n\n    if (match.substr(1) === change.substr(1)) {\n      changes.push(change);\n      state.index++;\n    } else {\n      conflicted = true;\n    }\n  }\n\n  if ((matchChanges[matchIndex] || '')[0] === '+'\n      && contextChanges) {\n    conflicted = true;\n  }\n\n  if (conflicted) {\n    return changes;\n  }\n\n  while (matchIndex < matchChanges.length) {\n    merged.push(matchChanges[matchIndex++]);\n  }\n\n  return {\n    merged,\n    changes\n  };\n}\n\nfunction allRemoves(changes) {\n  return changes.reduce(function(prev, change) {\n    return prev && change[0] === '-';\n  }, true);\n}\nfunction skipRemoveSuperset(state, removeChanges, delta) {\n  for (let i = 0; i < delta; i++) {\n    let changeContent = removeChanges[removeChanges.length - delta + i].substr(1);\n    if (state.lines[state.index + i] !== ' ' + changeContent) {\n      return false;\n    }\n  }\n\n  state.index += delta;\n  return true;\n}\n"]} |
|
368
|
|
|
|